# Introduction to SystemVerilog & Advanced Testbench

Lecturer: Jui-Huang Tsai



### **Outline**

- **✓** Section 1 Introduction
- ✓ Section 2 Design using SystemVerilog
  - Data Type, enumerate, structure, union
  - Procedure Block
  - Interface
- ✓ Section 3 Verification using SystemVerilog
  - OOP (Object-Oriented Programming)
  - Randomization



### **Outline**

- **✓** Section 1 Introduction
- **✓** Section 2 Design using SystemVerilog
  - Data Type, enumerate, structure, union
  - Procedure Block
  - Interface
- **✓ Section 3** Verification using SystemVerilog
  - OOP (Object-Oriented Programming)
  - Randomization



# Why SystemVerilog?



- ✓ Human factors may limit design more than technology
- ✓ Keys to solve the productivity crisis
  - Higher abstract language to enhance design & verification
  - Design techniques: hierarchical design, SoC design (IP reuse, platform-based design), etc.
  - CAD: algorithms & methodology

Source: http://access.ee.ntu.edu.tw/course/under\_project\_1011/slides/20120925\_IC\_Design\_Flow.pdf



# Why SystemVerilog?

### ✓ Designs efficiency

- Need to code for reuse and higher abstraction
- Need more efficient coding constructs with native language support

### **✓** System level hardware design/verification languages

 Unification of both syntax and semantics with one language improves communication between design team and verification team

### ✓ Testbenches are growing exponentially

- Find a way to generate more useful test data and make testing faster
- More advanced verification techniques (ex: OOP)
- Verification Library (ex: UVM, Universal Verification Methodology)



# What is SystemVerilog?

- ✓ The third generation of Verilog
  - Specified in IEEE Std 1800-2005
- ✓ Fully backward compatible with Verilog 2001





# Overview of SystemVerilog





# Overview of SystemVerilog





# The SystemVerilog Design Environment

#### √ testbed.sv

- Connecting testbench and design modules
- Generating clock
- Dump waveform

### √ design.sv

Design under test (DUT)

### ✓ pattern.sv

- Pattern
- Test program
- Assertion
- Coverage



Top level



# Simple Example: 8-bit Adder (1/3)

### **✓** RTL description



```
module adder(
      input logic [7:0] opa,opb,
      output logic [7:0] sum, output logic ca,
      input logic, clk, reset);
logic [8:0] result;
always_ff@(posedge clk , negedge reset)
begin:p_ADDER_BLOCK
      if(!reset) begin
            ca \le 0;
            sum <= 0;
      end
      else begin
            ca <= result [8];
            sum <= result [7:0];
      end
end: p_ADDER_BLOCK
always_comb
      result = opa + opb;
endmodule
```



# Simple Example: 8-bit Adder (2/3)

### **✓** Test pattern

- Generate stimulus
- Monitor response

```
program automatic test(output logic [7:0] opa,
                  output logic [7:0] opb,
                  output logic reset,
                  input wire clock,
                  input logic [7:0] sum,
                  input logic ca);
initial begin
      reset <= 1'b1;
      opa \leq = 0;
      opb <=0;
      repeat(5) @ (posedge clock);
      reset <= 1'b0;
      repeat(5) @ (posedge clock);
      reset <= 1'b1:
      opa<= 10;
      opb<= 20;
      @(posedge clock);
      d = d = d , opa, opb, sum);
      repeat(5) @(posedge clock);
      $display("Adder Simulation End");
end
endprogram
```



# Simple Example: 8-bit Adder (3/3)

## **✓** Top Level Harness file

```
`timescale 1ns/100ps
module top;
 parameter simulation cycle = 100;
 bit SystemClock;
 logic [7:0] opa, opb, sum;
                                                 instance testbench
 logic
           reset,ca;
test test_p(.opa (opa), .opb(opb), .clock(SystemClock), .reset(reset), .sum(sum), .ca
 adder dut( .opa(opa),.opb(opb),.clock(SystemClock),.reset(reset),.sum(sum),.ca(ca) );
 initial begin
                                                    instance design
  SystemClock = 0;
 forever begin
                                            Generate clock
  #(simulation_cycle/2)
   SystemClock = ~SystemClock;
 end
 end
                                            Dump waveform
 initial begin
     $fsdbDumpfile();
end
endmodule
```

Clock generated example:
forever #5ns clock = ~clock;



### **Outline**

- **✓** Section 1 Introduction
- ✓ Section 2 Design using SystemVerilog
  - Data Type, enumerate, structure, union
  - Procedure Block
  - Interface
- **✓ Section 3** Verification using SystemVerilog
  - OOP (Object-Oriented Programming)
  - Randomization



### **Outline**

- **✓** Section 1 Introduction
- ✓ Section 2 Design using SystemVerilog
  - Data Type, enumerate, structure, union
  - Procedure Block
  - Interface
- **✓ Section 3** Verification using SystemVerilog
  - OOP (Object-Oriented Programming)
  - Randomization



# Data Type in Verilog

- **✓** The Verilog language uses the reg type as a general purpose variable
  - Both combinational and sequential circuits can be model using reg
  - reg type has no connection to register, which often causes misunderstanding
- ✓ Lacks of user-defined type



# Data Type in SystemVerilog

### √ 4-State Variables (1|0|Z|X)

(default value is X if not initialized)

- logic w; // can be used in both assignment and procedure blocks
  - No input and output restriction anymore
  - No continuous or block procedure restriction
- reg r; // the same as logic
- integer i; // Verilog-2001 >= fixed size (32-bit data type)

#### √ 2-State Variables (1|0)

(Default value is 0 if not initialized)

```
shortint s; // 16-bit signed integer
```

- int i; // 32-bit signed integer
- longint I; // 64-bit signed integer
- byte b8;// 8-bit signed integer
- bit b; // 1-bit integer

```
logic net_A, net_B;

assign net_A = y;

assign net_A = y;

assign net_B = y;

net_B = y;

end
```



# **Synthesis About Data type**

- ✓ The bit, byte, shortint, int and longint types only store 2-state values. Synthesis treats these types as a 4-state reg variable with a corresponding vector size.
- ✓ There is a risk of a functional mismatch between simulation and the synthesized implementation, because synthesis does not maintain the 2-state behavior.
- ✓ Recommendation Use logic for almost all declarations. Avoid all 2state types in RTL models. These types can hide design problems.
  - Can lead to simulation vs. synthesis mismatches
- ✓ The one exception is to use an int variable for the iterator variable in forloops.



# **Array**

### **✓** Verilog-2001:

- A dimension declared before the object name is referred to as the "vector width" dimension.
- The dimensions declared after the object name are referred to as the "array" dimensions.
   reg [7:0] r1 [1:256]; //[7:0] is the vector width,[1:256] is the array size

#### **✓** SystemVerilog :

- "packed array" to refer to the dimensions declared before the object name (what Verilog-2001 refers to as the vector width)
- "unpacked array" is used to refer to the dimensions declared after the object name

```
bit [7:0] c1; // packed array
real u [7:0]; // unpacked array
```

Example

```
bit [3:0] [7:0] test [1:10]; // 10 entries of 4 bytes (packed into 32 bits)
// can be used as follows:
test [9] = test[8] + 1; // 4 byte add
test [7][3:2] = test [6][1:0]; // 2 byte copy
```



# **Typedef**

#### **✓** Make code clear

- SystemVerilog's data type system allows you to define quite complex types. To make this
  kind of code clear, the typedef facility was introduced
- typedef allows users to create their own names for type definitions that they will use frequently in their code

#### **✓** Example1:

```
typedef reg [7:0] octet;

cotet b;

// is the same as

reg [7:0] b;
```

#### **✓** Example2:

```
typedef octet [3:0] quadOctet;
quadOctet qBytes [1:10];
// is the same as
reg [3:0][7:0] qBytes [1:10];
```



### Structure

- ✓ Group related signals to enhance readability and clearly convey designer's intent (ex: CPU's instruction)
- **✓** Create structures data types:
  - Structures are a collection of variables

```
typedef struct{<data_type> variable0;<data_type> variable1;}struct_type;
```

#### **✓** Create structure variables:

```
struct{<data_type> variable0;<data_type> variable1;}struct_variable;
```



### Structure

#### **✓** Example:

```
typedef struct{
   int a,b;
   opcode_t opcode;
   logic[31:0] address;
   bit error;
} Instruction;

Instruction IR;
IR.address = 32'hF000001E;
```

```
1  struct{
2    int a,b;
3    opcode_t opcode;
4    logic[31:0] address;
5    bit error;
6  }IR;
7
8  IR.address = 32'hF000001E;
9
```

logic [6:0] Score;

```
typedef struct{
   Score Chinese[0:2];
   Score Math[0:2];
   Score English[0:2];
} student;
// Three Exams
```

Student A1; A1.Chinese[0] = 90; A1.Math[1] = 87;

| A1's score | 0  | 1  | 2 |
|------------|----|----|---|
| Chinese    | 90 | X  | X |
| Math       | X  | 87 | Х |
| English    | X  | X  | X |



### **Enumerate**

### **✓** Create enumerated data types:

- Defines a set of named values, which provides <u>built-in assertion!!</u>
- Data type defaults to *int* (32-bit, 2state, signed int)
- Variable initialized to 0 if initial values aren't specified

#### **✓** Create enum variables:

```
// anonymous enumerated type
```

- enum <type> {lists\_of\_enumerations} <enumvar>;
 enum { circle, ellipse, freeform } c;
// typed enumerated type

- typedef enum <type> {lists\_of\_enumerations} <enumtype>; enumtype variable\_name; typedef enum { circle, ellipse, freeform } ClosedCurve; ClosedCurve c;

### **✓** Often use to represent state machine



#### **Enumerate**

#### **✓** Example1:

```
// default int type
enum {red, yellow, green} light1, light2;
enum {bronze=3, silver, gold} medal;  // silver=4, gold=5
enum {a=0, b=7, c, d=8} alphabet;  // Syntax error

// enumerated type with a 2-bit logic type
typedef enum logic [1:0] {WAIT, LOAD, READY} state_t;
state_t state, next_state;
```

### **✓** Example2:

```
// Must match the size of the base type (default to int)
enum logic [2:0] {WAIT=3'b001, LOAD=3'b010, READY=3'b100} state;

//syntax error!!! (Error enum value size)
enum {WAIT=3'b001, LOAD=3'b010, READY=3'b100} state;
```

#### **Example3:**

```
typedef enum logic [2:0]
{ a=3'd0, b=3'd1, c, d=3'd5 } // ok, c=2
{ a=3'd0, b=3'd1, c, d=3'd2 } // error!
{ a=3'd0, b=3'd1, c=3'd7, d } // error!
```



# **Union (1/2)**

✓ A SystemVerilog union allows a single piece of storage to be represented different ways using different named member types.

```
typedef logic [3:0] type_A;
typedef logic [7:0] type_B;
typedef type_B[1:0] type_C;
typedef struct packed {
   integer temp int; // 32-bits
   logic[15:0] d;  // 16-bits
} type_strct;  // 64-bits
typedef union packed{
   type_strct
              mem_struct;
   type_A
              [15:0] mem_a;
              [7:0] mem_b;
   type_B
   type_C
              [3:0] mem c;
}my_union_type;
```

```
my_union_type my_union;
type A
              get_a;
type_B
             get b;
always_ff@(posedge clk or negedge inf.rst_n) begin
    if(!inf.rst_n)begin
        get a <= 'b0;
    end
    else begin
        get_a <= my_union.mem a[0];</pre>
    end
end
always_ff@(posedge clk or negedge inf.rst_n) begin
    if(!inf.rst_n)begin
        get b <= 'b0;
    end
    else begin
        get_b <= my_union.mem_b[1];</pre>
    end
end
```



# **Union (2/2)**



Assume this signal called "D", we can use "D.A[0]" to access the data in D[1:0].



# Package

✓ To enable sharing a user-defined type definition across multiple modules,
SystemVerilog adds packages to the Verilog language

```
package definitions;

parameter VERSION = "1.1";

typedef enum {ADD, SUB, MUL} opcodes_t;

typedef struct {
   logic [31:0] a, b;
   opcodes_t opcode;
   } instruction_t;

endpackage
```



# Package in module

✓ When the case statement references the enumerated labels of ADD, SUB, and MUL, as well as the function multiplier, it will find the definitions of these names in the definitions package

```
package definitions;

parameter VERSION = "1.1";

typedef enum {ADD, SUB, MUL} opcodes_t;

typedef struct {
   logic [31:0] a, b;
   opcodes_t opcode;
} instruction_t;

endpackage
```

# Package in module

✓ When the case statement references the enumerated labels of ADD, SUB, and MUL, as well as the function multiplier, it will find the definitions of these names in the definitions package

```
module ALU
(input definitions::instruction t IW,
 input logic
                                    clock,
 output logic [31:0]
                                    result
  import definitions::*;
                         // wildcard import
  always comb begin
    case (IW.opcode)
      ADD : result = IW.a + IW.b;
      SUB : result = IW.a - IW.b;
      MUL : result = multiplier(IW.a, IW.b);
    endcase
  end
endmodule
```



### **Outline**

- **✓** Section 1 Introduction
- ✓ Section 2 Design using SystemVerilog
  - Data Type, enumerate, structure, union
  - Procedure Block
  - Interface
- **✓ Section 3** Verification using SystemVerilog
  - OOP (Object-Oriented Programming)
  - Randomization



# **Procedure Block in Verilog**

- √ In Verilog, there are 2 kinds of procedure block
  - initial
  - always



### **Procedure Block**

- ✓ In Verilog, always is a general purpose procedure block
  - Can be use to model testbench, combination circuits, sequential circuits...etc
  - Depend on context which is not intuitive
  - Cannot be placed inside program or class and other procedure blocks
- ✓ System Verilog adds three new logic specific processes to show designers intent:
  - always\_comb
  - always\_ff
  - always\_latch
- ✓ In program, we may use forever@ to replace always blocks



# always\_comb

✓ The always\_comb procedural block is used to indicate the intent to model combinational logic.





# always\_ff

✓ The always\_ff procedural block is used to indicate that the intent to model synthesizable sequential logic behavior.

#### **✓** Example:



# always\_latch

✓ The always\_latch procedural block is used to indicate that the intent to model combinational logic with latch in it.

#### **✓** Example:

```
- always_latch
   if (enable)
      a_latch = a;
```



### **Outline**

- **✓** Section 1 Introduction
- ✓ Section 2 Design using SystemVerilog
  - Data Type, enumerate, structure, union
  - Procedure Block, Design intent
  - Interface
- **✓ Section 3** Verification using SystemVerilog
  - OOP (Object-Oriented Programming)
  - Randomization



### Interface

- ✓ The interface encapsulate communication between design blocks, and between design and verification blocks.
  - Replacing a group of names by a single one
  - An interface is a named bundle of wires, similar to a struct, except that an interface is allowed as a module port, while a struct is not.







### **Example**

```
module moduleA ( input bit
                                           clk
                     , input logic
                                           ack
 5
                      , input logic
                                           ready
 6
                      , output logic
                                           send
 8
                      , output logic [31:0] data
 9
10
         ... // actual module definition here
11
      endmodule
12
    module moduleB ( input bit
13
                                           clk
14
                      , input logic
                                           send
15
                      , input logic [31:0] data
16
                      , output logic
17
                                           ack
18
                      , output logic
                                           ready
19
             );
20
         ... // actual module definition here
     endmodule
21
22
23
     module top;
24
25
        clockgen CLOCKGEN (clk); // the clock generator
26
27
        moduleA AA (clk, ack, ready, send, data);
28
        moduleB BB (clk, send, data, ack, ready);
29
     endmodule
30
31
```

```
∃interface intf AB;
     logic
                  ack:
     logic
                  ready;
 6
     logic
                  send:
 7
     logic [31:0] data;
      ... // actual interface definition here
 9
     endinterface
10
11
    module moduleA ( input bit clk
12
                     , intf AB intf1
13
             ) :
14
        ... // actual module definition here
15
     endmodule
16
    module moduleB ( input bit clk
18
                     , intf AB intf2
19
             );
20
        ... // actual module definition here
     endmodule
21
22
23
     module top;
24
25
        intf AB intf();
                           // the interface declaration
        clockgen CLOCKGEN (clk); // the clock generator
26
27
28
                              (clk)
        moduleA AA ( .clk
29
                     ,.intf1 (intf)
30
               );
        moduleB BB ( .clk
31
                              (clk)
32
                     ,.intf2 (intf)
33
               );
     endmodule
```



### Modports of interface

✓ Modports are used to define direction of signal inside interface.

```
pinterface INF(input bit clk);
                                                SystemClock;
23
         logic
                 reset;
24
        instr t IW;
                                                inf(SystemClock);
         data t OUT;
                                           PATTERN test p(
26
                                    44
                                             .clk
                                                      (SystemClock),
        modport PATTERN (
                                             .inf
                                                      (inf.PATTERN)
             output reset,
                                    46
29
             output IW,
30
             input OUT
                                           `ifdef RTL
31
                                    49
                                           ALU dut (
32
                                    50
                                             .clk
                                                      (SystemClock),
33
        modport DESIGN (
                                                      (inf.DESIGN)
                                             .inf
34
             input reset,
35
             input IW,
                                            endif
36
             output OUT
37
38
39
    endinterface
40
```

```
□module ALU(
        input clk,
25
        INF.DESIGN inf
26
   p// input instr t IW, output data t OUT, input clk, reset);
   // shortint temp;
    logic [16:0] temp;
   // logic [16:0] temp ttt;
    logic temp ovf;
32 // Output sequential logic
    always ff @ (posedge clk)
   □begin : p OUTPUT REG LOGIC
        if(inf.reset)
35
36 🛓
        begin
37
             inf.OUT.result<='0;
38
            inf.OUT.ovf<='0;
39
        end
40
        else
41 🖨
        begin
             inf.OUT.result<=temp[15:0];</pre>
42
43
             inf.OUT.ovf<=temp[16];
        end
    end : p OUTPUT REG LOGIC
```

INF.sv TESTBED.sv

DESIGN.sv



#### **Outline**

- **✓** Section 1 Introduction
- **✓** Section 2 Design using SystemVerilog
  - Data Type, enumerate, structure
  - Procedure Block, Design intent
  - Interface
- ✓ Section 3 Verification using SystemVerilog
  - OOP (Object-Oriented Programming)
  - Randomization



## Object-oriented programming

- **✓** Objects, instances of classes
- **✓** Contents of "objects"
  - Data field
  - Constructor
  - Methods
- **✓** Each object should do what it should do, rather than several roles.
- ✓ Interaction between object and the object should be reduced as much as possible.





### Object-oriented programming

```
∃class Transcation
         //number of objects created
        static int count = 0;
        //unique instance ID
        int id;
        logic [31:0] addr, crc, data[8];
         function new();
            id = count++;
10
         endfunction
11
12
         function void display id;
13
            $display("%d",id);
        endfunction
14
15
16 Dendclass
```

```
□program test;
         /* other codes ...*/
 4
 5
         Transcation t;
         initial begin
             repeat (n) begin
                 t = new();
 8
 9
                 t.addr = $random();
                 transmit(t); // user defined task
10
11
             end
12
         end
13
         /* other codes ...*/
14
15
16 ⊟endprogram
```



#### **Outline**

- **✓** Section 1 Introduction
- **✓** Section 2 Design using SystemVerilog
  - Data Type, enumerate, structure
  - Procedure Block, Design intent
  - Interface
- ✓ Section 3 Verification using SystemVerilog
  - OOP (Object-Oriented Programming)
  - Randomization



### **How To Do Verification?**

- ✓ How long will simulation run if exhaustive testing of a 32-bit adder is required?
  - Assume that one set of input and output can be verified every 1ns.



A day?

A week?

A year?



### **Alternatives to Exhaustive Verification?**

- ✓ What is the goal of verification if exhaustive verification is unachievable?
- ✓ Verify with sufficient set of vectors to gain a level of confidence that product will ship with a tolerable field-failure rate.
- **✓** The best known mechanism is randomization:
  - Randomization of data
  - Use constraints to narrow the scope



### Randomization

- **✓** Two types of random properties are supported:
  - rand
  - randc
- **✓** rand properties can assume any legal value:
  - Values can repeat without exhausting all possible values
- **✓** randc properties can be up to 16 bits:
  - Exhaust all values before repeating any individual value
  - For non-repeating bit values > 16 bits, use concatenation
- √ rand and randc properties are randomized when the class method randomize() is called:
  - randomize() is built-in with every class
  - 1 is returned if successful, 0 if randomization failed



### Randomization Example

#### **Ex1**:

```
class random interval;
        rand int interval ;
        function new ( int seed );
                this.srandom(seed);
        endfunction
        constraint limit
                interval inside{1 , 2 , 3};
endclass
random interval interval rand = new(1) ;
int i ;
initial
begin
        i = interval_rand.randomize() ;
        if(i == 0 ) $display("ERROR");
end
```

#### **Ex2**:



#### Constraints

### **✓** Class properties are constrained in a constraint block

- Use operator or distribution constraints
- Arrays can be constraint with functions like size()

### **✓** For distribution specification:

Distributed over a specified range with keyword inside:

```
constraint Limit1 {
class member 
Sainside { [5:7], 10, 15 };
// 5,6,7,10,15 equally weighted probability
}
```

Excluded from a specified range with ! operator:

```
constraint Limit2 {
!( sa inside { [1:10], 15 } );
// not 1 through 10 or 15
}
```



### **Appendix: Constraints**

- **✓** Operator Constraints
- **✓** Implication operators:

```
- ->
- if(...)...[else]
```

```
typedef enum { low, mid, high } AddrType;
class MyBus;
rand bit[7:0] addr;
rand AddrType atype;
constraint addr_range {
(atype == low) \rightarrow addr inside { [0:15] };
(atype == mid ) -> addr inside { [16:127] };
(atype == high) -> addr inside { [128:255] };
// same as:
// if (atype == low) addr inside { [0:15] };
// if (atype == mid) addr inside { [16:127] };
// if (atype == high) addr inside { [128:255] };
endclass
```



## **Appendix: Program Block**

- **✓** Purpose: Identifies verification code
- **✓** A program differs from a module
  - Only initial blocks allowed
  - Execute in reactive region





## **Appendix: Program Block**





### **Appendix: Program Block**

```
module DUT();
reg q = 0;
reg clk = 0;
initial
#10 clk = 1;
always @(posedge clk)
q \le 1;
endmodule
module Module_based_TB();
always @ (posedge DUT.clk) $display("Module_based_TB : q = %b\n", DUT.q);
endmodule
program Program_based_TB();
initial
forever @(posedge DUT.clk) $display("Program_based_TB : q = %b\n", DUT.q);
endprogram
RESULT:
Module_based_TB: q = 0
program_based_TB: q = 1
```



# **Appendix: Generate**

### **✓** Useful for vector or array

```
genvar i;
     logic [1:0] temp test [0:3];
    ⊟generate
          for (i=0; i<4; i=i+1)
          begin:test for
10
              always ff @ (posedge clk)
              begin
              if (inf.reset)
                  temp test[i] <= 0;
              else
14
15
                  temp test[i] <= i;</pre>
16
              end
17
          end
18
19
     endgenerate
```



# Appendix: More about randomize()

### **✓** When randomize() executes, three events occur:

- pre\_randomize() is called
- Randomization is executed
- post\_randomize() is called

### √ pre\_randomize()

- Optional
- Set/Correct constraints before randomization

### √ post\_randomize()

- Optional
- Make corrections after randomization

```
function void pre_randomize ();
   $display ("This will be called just before randomization");
endfunction

function void post_randomize ();
   $display ("This will be called just after randomization");
endfunction
```



#### Reference

- ✓ ASIC World (<a href="http://www.asic-world.com/systemverilog/index.html">http://www.asic-world.com/systemverilog/index.html</a>)
- √ electroSofts (<a href="http://electrosofts.com">http://electrosofts.com</a>)
- √ AsicGuru.com (<a href="http://www.asicguru.com/home/6/">http://www.asicguru.com/home/6/</a>)
- ✓ Verification Guide (<a href="http://www.verificationguide.com/p/systemverilog-tutorial.html">http://www.verificationguide.com/p/systemverilog-tutorial.html</a>)
- ✓ Project VeriPage (<a href="http://www.project-veripage.com/index.php">http://www.project-veripage.com/index.php</a>)
- ✓ Language Reference Manual 
  (<a href="http://www.ece.uah.edu/~gaede/cpe526/SystemVerilog\_3.1a.pdf">http://www.ece.uah.edu/~gaede/cpe526/SystemVerilog\_3.1a.pdf</a>)

